JSON Schema in Practice - Advanced Topics
In the previous tutorial, we have learned the basics of JSON Schema. In this tutorial, we will continue with more advanced topics such as generic keywords, schema composition and schema conditionality.
Generic Keywords
Some of the annotation keywords used in JSON Schema are:
- title
- description
- default.
None of these annotation words are required but enhance good practice for self-documenting Schemas. The title and description keywords must be strings. The default keyword specifies a default value.
The style of the JSON Schema:
{
"title": "Match anything",
"description": "Description of the Schema.",
"default": "Default value"
}
Enumerated Values
The enum keyword is used to restrict a value to a fixed set of values. It must be an array with at least one element, where each element is unique.
An example JSON Schema with enum keyword:
{
"enum": ["red", "green"]
}
This schema accepts:
"red"
"green"
but does not accept:
"blue"
Constant Values
The const keyword is used to restrict a value to a single value.
An example JSON Schema with the const keyword:
{
"properties": {
"country": {
"const": "USA"
}
}
}
This schema only accepts:
{ "country": "USA" }
and does not accept:
{ "country": "Canada" }
{ "country": "Germany" }
Schema Composition
JSON Schema includes a few keywords for combining Schemas together. The most used ones are allOf, anyOf, oneOf, and not. All of these keywords must be set to an array, where each item is a Schema.
- allOf: (AND) Must be valid against all of the subSchemas
- anyOf: (OR) Must be valid against any of the subSchemas
- oneOf: (XOR) Must be valid against exactly one of the subSchemas
allOf
To validate allOf, the given data must be valid against all of the given subSchemas.
An example JSON Schema defined with the keyword allOf:
{
"allOf": [{ "type": "string" }, { "maxLength": 7 }]
}
This schema accepts this as it is both a string and its length does not excede 7:
"short"
But does not accept this as its length is larger than 7:
"long string"
anyOf
To validate anyOf, the given data must be valid against any (one or more) of the given subSchemas.
An example JSON Schema with the keyword anyOf:
{
"anyOf": [
{ "type": "string", "maxLength": 5 },
{ "type": "number", "minimum": 0 }
]
}
It is enough when only one of these criteria is satisfied.
So this one is accepted as it is string with the length of 5:
"short"
And also this one is accepted because it is a number and larger than 0:
"12"
But this one is not accepted since it does not fit none of the criteria:
"long string"
oneOf
To validate oneOf, the given data must be valid against exactly one of the given subSchemas.
An example JSON Schema with the keyword oneOf:
{
"oneOf": [
{ "type": "number", "multipleOf": 5 },
{ "type": "number", "multipleOf": 2 }
]
}
This schema will accept numbers which are either multiple of 5, or 2. It will perform an XOR operation.
It accepts this one as it is multiple of 5:
25
and this as it is multiple of 2:
4
however this one is not accepted because it is both multiple of 2 and 5:
10
not
The not keyword declares that an instance validates if it doesn’t validate against the given subSchema.
An example JSON Schema with the not keyword:
{
"not": { "type": "string" }
}
It will accept this as the type is number:
25
And this as it is type of boolean:
true
But this is not accepted since it is type string:
"10"
Schema Conditionality
The if, then and else keywords allow the application of a subSchema based on the outcome of another Schema, as it is in other programming languages.
If it is valid, then must also be valid (and else is ignored). If it is invalid, else must also be valid (and then ignored).
Here is an example JSON Schema defined with the keywords if, then, else:
{
"type": "object",
"if": {
"properties": {
"country": { "const": "USA" }
}
},
"then": {
"properties": {
"nationality": { "const": "American" }
}
},
"else": {
"properties": {
"nationality": { "const": "Canadian" }
}
}
}
This schema will accept:
{
"country": "USA",
"nationality": "American"
}
And does not accept:
{
"country": "Canada",
"nationality": "American"
}